File: //opt/alt/python37/lib/python3.7/site-packages/clcommon/cpapi/plugins/directadmin.py
# -*- coding: utf-8 -*-
"""
CloudLinux API for DirectAdmin control panel
"""
from __future__ import absolute_import
from __future__ import division
from __future__ import print_function
import glob
import os
import re
import sys
import json
import subprocess
from typing import List, Tuple, Dict # NOQA
from clcommon.clpwd import ClPwd
from clcommon.clconfpars import load as loadconfig, load_fast, apache_conf_parser, nginx_conf_parser, \
WebConfigParsingError, WebConfigMissing
from clcommon.cpapi.cpapiexceptions import NoDBAccessData, CpApiTypeError, NoDomain, ReadFileError, ParsingError
from clcommon.cpapi.plugins.universal import get_admin_email as universal_get_admin_email
from clcommon.utils import exec_utility
from clcommon.cpapi.GeneralPanel import GeneralPanelPluginV1
__cpname__ = 'DirectAdmin'
DA_DIR = '/usr/local/directadmin'
DA_CONF = os.path.join(DA_DIR, 'conf/directadmin.conf')
DA_DATA_DIR = os.path.join(DA_DIR, 'data')
DA_DB_CONF = os.path.join(DA_DIR, 'conf/mysql.conf')
DA_USERS_PATH = os.path.join(DA_DATA_DIR, 'users')
DA_OPT_PATH = os.path.join(DA_DIR, 'custombuild', 'options.conf')
USER_CONF = 'user.conf'
DOMAINOWNERS = '/etc/virtual/domainowners'
ADMIN_DIR = os.path.join(DA_DATA_DIR, 'admin')
RESELLERS_LIST = os.path.join(ADMIN_DIR, 'reseller.list')
ADMINS_LIST = os.path.join(ADMIN_DIR, 'admin.list')
USER_PATTERN = re.compile(r'.+/(.+)/%s' % re.escape(USER_CONF))
DA_BINARY = '/usr/local/directadmin/directadmin'
# Domain to user map (from /etc/virtual/domainowners)
# For example: {'user1.com': 'user1', 'sub.user1.com': 'user1' .... }
_domain_to_user_map = None
# WARN: Probably will be deprecated for our "official" plugins.
# See pluginlib.detect_panel_fast()
def detect():
return os.path.isfile('/usr/local/directadmin/directadmin') or \
os.path.isfile('/usr/local/directadmin/custombuild/build')
def db_access():
access = dict()
try:
login_data = loadconfig(DA_DB_CONF)
access['login'] = login_data['user']
access['pass'] = login_data['passwd']
except IOError as err:
raise NoDBAccessData('Can not open file with data to database access; ' + str(err))
except KeyError:
raise NoDBAccessData('Can not get database access data from file %s' % (DA_DB_CONF,))
return access
def cpusers():
match_list = [USER_PATTERN.match(path) for path in glob.glob(os.path.join(DA_USERS_PATH, '*', USER_CONF))]
users_list = [match.group(1) for match in match_list if match]
return tuple(users_list)
def resellers():
stream = open(RESELLERS_LIST)
resellers_list = [item.strip() for item in stream.readlines()]
stream.close()
return tuple(resellers_list)
def admins():
stream = open(ADMINS_LIST)
admins_list = [item.strip() for item in stream.readlines()]
stream.close()
return set(admins_list)
def dblogin_cplogin_pairs(cplogin_lst=None, with_system_users=False):
from clcommon.cpapi.plugins.universal import _dblogin_cplogin_pairs
access = db_access()
data = _dblogin_cplogin_pairs(cplogin_lst=cplogin_lst, access=access)
if with_system_users:
data += tuple(get_da_user(DA_USERS_PATH).items())
return data
def get_da_user(path, quiet=True):
users = {}
cur_dir = os.getcwd()
os.chdir(path)
dir_list = glob.glob('./*')
for user_dir in dir_list:
if os.path.isdir(user_dir):
file_domains = path + '/' + user_dir + '/domains.list'
try:
f = open(file_domains)
if len(f.readline()) > 0:
user_name = user_dir[2:]
users[user_name] = user_name
f.close()
except IOError:
if not quiet:
sys.stderr.write("No file " + file_domains)
os.chdir(cur_dir)
return users
def cpinfo(cpuser=None, keyls=('cplogin', 'package', 'mail', 'reseller', 'dns', 'locale'), search_sys_users=True):
returned = list()
if isinstance(cpuser, str):
cpusers_list = [cpuser]
elif isinstance(cpuser, (list, tuple)):
cpusers_list = tuple(cpuser)
elif cpuser is None:
cpusers_list = cpusers()
else:
raise CpApiTypeError(funcname='cpinfo', supportedtypes='str|unicode|list|tuple',
received_type=type(cpuser).__name__)
def _get_reseller(config):
if config.get('usertype') == 'reseller':
return config.get('username')
return config.get('creator')
_user_conf_map = {'cplogin': lambda config: config.get('username'),
'package': lambda config: config.get('package'),
'mail': lambda config: config.get('email'),
'reseller': lambda config: _get_reseller(config),
'dns': lambda config: config.get('domain'),
'locale': lambda config: config.get('language')}
keyls_ = [_user_conf_map[key] for key in keyls]
for username in cpusers_list:
user_conf_file = os.path.join(DA_USERS_PATH, username, USER_CONF)
if os.path.exists(user_conf_file):
user_config = load_fast(user_conf_file)
returned.append([key(user_config) for key in keyls_])
return returned
def _get_domains_docroots_from_da_binary():
"""
Get domains and subdomains with its document root directories.
Data source - DA native binary
:return: Data dictionary or {} if can't receive data from DA binary
Data dictionary example:
{u'admin': {},
u'clnodetest': {u'cloudtestnode.com': u'/home/clnodetest/domains/cloudtestnode.com/public_html'},
u'clresuser2': {u'2.22': u'/home/clresuser2/domains/2.22/public_html'},
u'clres1': {u'aaaa.1': u'/home/clres1/domains/aaaa.1/public_html',
u'res1.aaaa.1': u'/home/clres1/domains/aaaa.1/public_html/res1'}
}
"""
_users_domains_docroot_map = dict()
try:
_, da_out = exec_utility(DA_BINARY, ['--DocumentRoot'])
users_domains_data = json.loads(da_out)
# domains_data example:
# {u'users': {u'admin': {u'domains': {}},
# u'normuser1': {u'domains': {
# u'user.domain': {u'subdomains': {},
# u'public_html': u'/home/normuser1/domains/user.domain/public_html',
# u'private_html': u'/home/normuser1/domains/user.domain/private_html'}
# }
# },
# u'cltest1': {u'domains': {
# u'cltest1.com': {u'subdomains': {
# u'mk': {u'public_html': u'/home/cltest1/domains/cltest1.com/public_html/mk',
# u'private_html': u'/home/cltest1/domains/cltest1.com/private_html/mk'}
# },
# u'public_html': u'/home/cltest1/domains/cltest1.com/public_html',
# u'private_html': u'/home/cltest1/domains/cltest1.com/private_html'}
# }
# }
# }
# }
# Retrive users dictionary
users_data_dict = users_domains_data['users']
# Parse users data and fill result dictionary
for user_name, user_domains_dict in users_data_dict.items():
domains_data = user_domains_dict['domains']
user_data = dict()
for domain_name, domain_v in domains_data.items():
subdomains_dict = domain_v['subdomains']
domain_docroot = domain_v['public_html']
user_data[domain_name] = domain_docroot
# Process subdomains
for subdomain_name, subdomain_data in subdomains_dict.items():
user_data['%s.%s' % (subdomain_name, domain_name)] = subdomain_data['public_html']
_users_domains_docroot_map[user_name] = user_data
except (OSError, IOError, TypeError, KeyError, IndexError, ValueError):
# Errors: DA_BINARY access denined, non-JSON output, invalid JSON format
pass
return _users_domains_docroot_map
def _docroot_under_root(domain):
# type: (str) -> Tuple[str, str]
"""
Old method for getting doc_root for domain under root
Method reads DA config
:return: (doc_root, username) cortege
"""
user_name = None
# Load /etc/virtual/domainowners
_load_domains_owners()
# Find supposed owner of domain
for main_domain in list(_domain_to_user_map.keys()):
if domain == main_domain or domain.endswith('.%s' % main_domain):
# Parent domain found
user_name = _domain_to_user_map[main_domain]
break
if user_name is None:
domains_list = list()
else:
domains_list = userdomains(user_name)
for d in domains_list:
if domain in d:
return d[1], user_name
def _docroot_under_user_old_mechanism(domain):
# type: (str) -> Tuple[str, str]
"""
Old method for getting doc_root for domain under user
Method parses /home/<username>/domains directory
:return: (doc_root, username) cortege
"""
clpwd = ClPwd()
user_pw = clpwd.get_pw_by_uid(os.getuid())[0]
list_domains_and_doc_roots = _get_domains_list_as_user(user_pw.pw_dir)
for domain_data in list_domains_and_doc_roots:
if domain_data['server_name'] == domain:
return domain_data['document_root'], user_pw.pw_name
def docroot(domain):
# type: (str) -> Tuple[str, str]
"""
Retrieves document root for domain
:param domain: Domain to determine doc_root
:return: Cortege: (doc_root, domain_user)
"""
domain = domain.strip()
# DA binary collects ALL info about users if we drop permissions using os.seteuid and stay uid == 0.
# And DA binary collects info about one user if it's run into CageFS.
# So we use uid instead euid
uid = os.getuid()
euid = os.geteuid()
# Method _get_domains_docroots_from_da_binary is very slow under root,
# because calling `directadmin --DocumentRoot` collects info about
# domains and doc_roots for all users on server, and this takes a lot of time
if euid != 0:
if uid != 0:
# Try to get users/domains data from DA binary for one user,
# because we are calling DA binary under user
_users_domains_docroot_map = _get_domains_docroots_from_da_binary()
if _users_domains_docroot_map:
for username, user_domains_data_dict in _users_domains_docroot_map.items():
if domain in user_domains_data_dict:
# Needed domain found
return user_domains_data_dict[domain], username
# We using old method in case if DA binary doesn't have key `--DocumentRoot`
else:
return _docroot_under_user_old_mechanism(domain)
else:
return _docroot_under_user_old_mechanism(domain)
else:
return _docroot_under_root(domain)
raise NoDomain("Can't obtain document root for domain '%s'" % domain)
def _is_nginx_installed():
"""
Check if nginx is installed via custombuild;
"""
config = loadconfig(DA_CONF)
return bool(int(config.get('nginx', 0)) or int(config.get('nginx_proxy', 0)))
def _get_domains_list_as_root(user_path):
"""
Get domains list for user from httpd or nginx config as root
:param user_path: path to DA directory of user's profile
:return: parsed httpd or nginx config
:rtype: list
"""
try:
if _is_nginx_installed():
httpd_conf = nginx_conf_parser(os.path.join(user_path, 'nginx.conf'))
else:
httpd_conf = apache_conf_parser(os.path.join(user_path, 'httpd.conf'))
except WebConfigParsingError as e:
raise ParsingError(e.message)
except WebConfigMissing:
return []
return httpd_conf
def _get_domains_list_as_user(user_home):
# type: (str) -> List[Dict[str, str, bool]]
"""
Get domains list for user from ~/domains directory as user.
Method DOESN'T search subdomains, because it's almost impossible detect by user's
folders without privileges escalation
:param user_home: path to user home
:return: list of dictionaries {'server_name': 'domain', 'document_root': 'doc_root', 'ssl': False}
"""
domains_dir = 'domains'
doc_root_dir = 'public_html'
domains_list = list()
domains_path = os.path.join(user_home, domains_dir)
# Searching main domains
# All directories of main domains are saved in ~/domains directory
for domain_dir in os.listdir(domains_path):
domain_path = os.path.join(domains_path, domain_dir)
doc_root_path = os.path.join(domains_path, domain_dir, doc_root_dir)
if os.path.isdir(domain_path) and os.path.isdir(doc_root_path):
domains_list.append({
'server_name': domain_dir,
'document_root': doc_root_path,
'ssl': False,
})
else:
continue
return domains_list
def userdomains(cpuser):
# type: (str) -> List[Tuple[str, str]]
"""
Get user's domains list
:return list: domain names
Example:
[('cltest1.com', '/home/cltest1/domains/cltest1.com/public_html'),
('mk.cltest1.com', '/home/cltest1/domains/cltest1.com/public_html/mk'),
('cltest11.com', '/home/cltest1/domains/cltest11.com/public_html')
]
"""
domains_list = list()
# DA binary collects ALL info about users if we drop permissions using os.seteuid and stay uid == 0.
# And DA binary collects info about one user if it's run into CageFS.
# So we use uid instead euid
uid = os.getuid()
euid = os.geteuid()
# Method _get_domains_docroots_from_da_binary is very slow under root,
# because calling `directadmin --DocumentRoot` collects info about
# domains and doc_roots for all users on server, and this takes a lot of time
if uid != 0:
# Try to get users/domains data from DA binary for one user,
# because we are calling DA binary under user
_users_domains_docroot_map = _get_domains_docroots_from_da_binary()
if cpuser in _users_domains_docroot_map:
# User present in data
domains_list = list(_users_domains_docroot_map[cpuser].items())
return domains_list
# old method to get list of user's domains
main_domain_path = ''
user_path = os.path.join(DA_USERS_PATH, cpuser)
if not os.path.exists(user_path):
return domains_list
user_home = os.path.expanduser('~' + cpuser)
public_path = os.path.join(user_home, 'public_html')
if os.path.exists(public_path) and os.path.islink(public_path):
main_domain_path = os.path.realpath(public_path)
if euid == 0:
httpd_conf = _get_domains_list_as_root(user_path)
else:
httpd_conf = _get_domains_list_as_user(user_home)
for domain in httpd_conf:
if domain['ssl'] is True:
continue
# Put main domain in start of list
if domain['server_name'] in main_domain_path:
domains_list.insert(0, (domain['server_name'], domain['document_root']))
else:
domains_list.append((domain['server_name'], domain['document_root']))
return domains_list
def homedirs():
"""
Detects and returns list of folders contained the home dirs of users of the DirectAdmin
:return: list of folders, which are parent of home dirs of users of the panel
"""
home_dirs = set()
clpwd = ClPwd()
users_dict = clpwd.get_user_dict()
for user_name, pw_user in list(users_dict.items()):
conf_file = os.path.join(DA_USERS_PATH, user_name, USER_CONF)
if os.path.exists(conf_file):
home_dir = os.path.dirname(pw_user.pw_dir)
home_dirs.add(home_dir)
return list(home_dirs)
def domain_owner(domain):
"""
Return domain's owner
:param domain: Domain/sub-domain/add-domain name
:return: user name or None if domain not found
"""
return _load_domains_owners().get(domain, None)
def _load_domains_owners():
# type: () -> dict[str, str]
"""Get domain<->user map from /etc/virtual/domainowners file"""
global _domain_to_user_map
if _domain_to_user_map:
return _domain_to_user_map
# 1. Load DA data file
try:
stream_ = open(DOMAINOWNERS)
domains_lines = stream_.readlines()
stream_.close()
except (OSError, IOError) as e:
_domain_to_user_map = None
raise ReadFileError(str(e))
# 2. File loaded successfully, parse data and fill dictionaries
_domain_to_user_map = dict()
for line_ in domains_lines:
line_ = line_.strip()
# pass empty line
if not line_:
continue
domain_, user_ = line_.split(':')
domain_ = domain_.strip()
user_ = user_.strip()
# Fill domain to user map
_domain_to_user_map[domain_] = user_
return _domain_to_user_map
def reseller_users(resellername):
"""
Return list of reseller users
:param resellername: reseller name; return empty list if None
:return list[str]: user names list
"""
if resellername is None:
return []
users_list_file = os.path.join(DA_USERS_PATH, resellername, 'users.list')
try:
with open(users_list_file) as users_list:
users_list = [item.strip() for item in users_list]
users_list.append(resellername)
return users_list
except (IOError, OSError):
return []
def reseller_domains(resellername=None):
"""
Get pairs user <=> domain for given reseller;
Empty list if cannot get or no users found;
:type resellername: str
:return list[tuple[str, str]]: tuple[username, main_domain]
"""
if resellername is None:
return []
users = reseller_users(resellername)
return dict(cpinfo(users, keyls=('cplogin', 'dns')))
def get_admin_email():
admin_user_file = os.path.join(DA_USERS_PATH, 'admin', USER_CONF)
cnf = loadconfig(admin_user_file)
return cnf.get('email', universal_get_admin_email())
def is_reseller(username):
"""
Check if given user is reseller;
:type username: str
:rtype: bool
:raise: ParsingError, ReadFileError
"""
user_config = os.path.join(DA_USERS_PATH, username, USER_CONF)
if os.path.exists(user_config):
try:
return loadconfig(user_config)['usertype'] == 'reseller'
except IndexError:
raise ParsingError('User config exists, but no usertype given')
return False
def get_user_login_url(domain):
return 'http://{domain}:2222'.format(domain=domain)
def is_reseller_limits_supported():
"""
if control panel is support reseller limits, method return True, else - False
:return: True of False
"""
return True
def _get_da_php_config():
"""
Return map (PHP_DA_CODE:{PHP_HANDLER, PHP_VERSION})
:return:
"""
_php_da_map = {}
try:
php_cfg = loadconfig(DA_OPT_PATH)
except (IOError, OSError):
return None
_php_da_map['1'] = {}
_php_da_map['1']['handler_type'] = php_cfg.get('php1_mode')
_php_da_map['1']['php_version_id'] = php_cfg.get('php1_release')
_php_da_map['2'] = {}
_php_da_map['2']['handler_type'] = php_cfg.get('php2_mode')
_php_da_map['2']['php_version_id'] = php_cfg.get('php2_release')
return _php_da_map
def _get_php_code_info_for_domain(domain, owner):
"""
Return php code from domain config
:param domain:
:param owner:
:return: string '1' or '2' - php code in DA
"""
domain_config_file = os.path.join(DA_USERS_PATH, str(owner), 'domains', str(domain) + '.conf')
try:
domain_config = loadconfig(domain_config_file)
except (IOError, OSError):
return '1'
domain_php = domain_config.get('php1_select')
# None - DA custombuild has only one php version
# '0' - it means that user selected default version PHP of DA custombuild
if domain_php is None or domain_php == '0':
domain_php = '1'
return domain_php
def get_domains_php_info():
"""
Return php version information for each domain
:return: domain to php info mapping
Example output:
{'cltest.com': {'handler_type': 'mod_php',
'php_version_id': '7.1',
'username': 'cltest'},
'cltest2.com': {'handler_type': 'fastcgi',
'php_version_id': '7.3',
'username': 'kek_2'},
'cltest3.com': {'handler_type': 'suphp',
'php_version_id': '5.5',
'username': 'cltest3'},
'omg.kek': {'handler_type': 'php-fpm',
'php_version_id': '5.2',
'username': 'cltest'}}
:rtype: dict[str, dict]
"""
map_domain_user = _load_domains_owners()
result_map = {}
php_da_map = _get_da_php_config()
if php_da_map is None:
return result_map
for domain, owner in map_domain_user.items():
php_info_code = _get_php_code_info_for_domain(domain, owner)
result_map[domain] = {'username': owner,
'handler_type': php_da_map[php_info_code]['handler_type'],
'php_version_id': php_da_map[php_info_code]['php_version_id']
}
return result_map
class PanelPlugin(GeneralPanelPluginV1):
def __init__(self):
super(PanelPlugin, self).__init__()
self.ADMINS_LIST = os.path.join(ADMIN_DIR, 'admin.list')
def getCPName(self):
"""
Return panel name
:return:
"""
return __cpname__
def get_cp_description(self):
"""
Retrieve panel name and it's version
:return: dict: { 'name': 'panel_name', 'version': 'panel_version', 'additional_info': 'add_info'}
or None if can't get info
"""
try:
p = subprocess.Popen(
['/usr/local/directadmin/directadmin', 'v'],
stdout=subprocess.PIPE,
stderr=subprocess.PIPE,
text=True
)
out, err = p.communicate()
return {'name': __cpname__, 'version': out.split('v.')[1].strip(), 'additional_info': None}
except:
return None
def db_access(self):
"""
Getting root access to mysql database.
For example {'login': 'root', 'db': 'mysql', 'host': 'localhost', 'pass': '9pJUv38sAqqW'}
:return: root access to mysql database
:rtype: dict
:raises: NoDBAccessData
"""
return db_access()
def cpusers(self):
"""
Generates a list of cpusers registered in the control panel
:return: list of cpusers registered in the control panel
:rtype: tuple
"""
return cpusers()
def resellers(self):
"""
Generates a list of resellers in the control panel
:return: list of cpusers registered in the control panel
:rtype: tuple
"""
return resellers()
def is_reseller(self, username):
"""
Check if given user is reseller;
:type username: str
:rtype: bool
"""
return is_reseller(username)
# unlike admins(), this method works fine in post_create_user
# hook; looks like directadmin updates admins.list a little bit later
# then calls post_create_user.sh
def is_admin(self, username):
"""
Return True if username is in admin names
:param str username: user to check
:return: bool
"""
user_conf_file = os.path.join(DA_USERS_PATH, username, USER_CONF)
if not os.path.exists(user_conf_file):
return False
user_config = load_fast(user_conf_file)
return user_config['usertype'] == 'admin'
def dblogin_cplogin_pairs(self, cplogin_lst=None, with_system_users=False):
"""
Get mapping between system and DB users
@param cplogin_lst :list: list with usernames for generate mapping
@param with_system_users :bool: add system users to result list or no.
default: False
"""
return dblogin_cplogin_pairs(cplogin_lst, with_system_users)
def cpinfo(self, cpuser=None, keyls=('cplogin', 'package', 'mail', 'reseller', 'dns', 'locale'),
search_sys_users=True):
"""
Retrieves info about panel user(s)
:param str|unicode|list|tuple|None cpuser: user login
:param keyls: list of data which is necessary to obtain the user,
the values​can be:
cplogin - name/login user control panel
mail - Email users
reseller - name reseller/owner users
locale - localization of the user account
package - User name of the package
dns - domain of the user
:param bool search_sys_users: search for cpuser in sys_users or in control panel users (e.g. for Plesk)
:return: returns a tuple of tuples of data in the same sequence as specified keys in keylst
:rtype: tuple
"""
return cpinfo(cpuser, keyls, search_sys_users=search_sys_users)
def get_admin_email(self):
"""
Retrieve admin email address
:return: Host admin's email
"""
return get_admin_email()
def docroot(self, domain):
"""
Return document root for domain
:param str|unicode domain:
:return Cortege: (document_root, owner)
"""
return docroot(domain)
def userdomains(self, cpuser):
"""
Return domain and document root pairs for control panel user
first domain is main domain
:param str|unicode cpuser: user login
:return list of tuples (domain_name, documen_root)
"""
return userdomains(cpuser)
def homedirs(self):
"""
Detects and returns list of folders contained the home dirs of users of the cPanel
:return: list of folders, which are parent of home dirs of users of the panel
"""
return homedirs()
def reseller_users(self, resellername=None):
"""
Return reseller users
:param resellername: reseller name; autodetect name if None
:return list[str]: user names list
"""
return reseller_users(resellername)
def reseller_domains(self, reseller_name=None):
"""
Get dict[user, domain]
:param reseller_name: reseller's name
:rtype: dict[str, str|None]
:raises DomainException: if cannot obtain domains
"""
return reseller_domains(reseller_name)
def get_user_login_url(self, domain):
"""
Get login url for current panel;
:type domain: str
:rtype: str
"""
return get_user_login_url(domain)
def admins(self):
"""
List all admins names in given control panel
:return: list of strings
"""
return admins()
def panel_features(self):
"""
Retrives dictionary with different panel properties
:return: dictionry
"""
return {'reseller_limits_supported': is_reseller_limits_supported()}
def domain_owner(self, domain):
"""
Return domain's owner
:param domain: Domain/sub-domain/add-domain name
:rtype: str
:return: user name or None if domain not found
"""
return domain_owner(domain)
def get_domains_php_info(self):
"""
Return php version information for each domain
:return: domain to php info mapping
:rtype: dict[str, dict]
"""
return get_domains_php_info()
@staticmethod
def _get_da_skin_name():
"""
Retrieve current DA skin name
:return: Current DA skin name. None if unknown
"""
# grep '^docsroot=' /usr/local/directadmin/conf/directadmin.conf | cut -d/ -f4
config = loadconfig(DA_CONF)
docsroot = config.get('docsroot', None)
# docsroot like './data/skins/evolution'
if docsroot is None:
return None
return docsroot.split('/')[-1]
@staticmethod
def get_encoding_name():
"""
Retrieve encoding name, used for package/reseller names
:return:
"""
enhanced_skin_config = os.path.join(DA_DIR, "data/skins/enhanced/lang/en/lf_standard.html")
default_encoding = 'utf8'
current_skin = PanelPlugin._get_da_skin_name()
if current_skin == 'enhanced':
# For enchanced skin we read encoding from its config
# :LANG_ENCODING=iso-8859-1 see LU-99 for more info
skin_config = loadconfig(enhanced_skin_config)
# Option in file is 'LANG_ENCODING', but key is lowercase
return skin_config.get('lang_encoding', default_encoding)
return default_encoding